New Case Study:See how Anthropic automated 95% of dependency reviews with Socket.Learn More
Socket
Sign inDemoInstall
Socket

@travetto/di

Package Overview
Dependencies
Maintainers
1
Versions
316
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

@travetto/di

Dependency registration/management and injection support.

  • 0.6.8
  • Source
  • npm
  • Socket score

Version published
Weekly downloads
623
increased by335.66%
Maintainers
1
Weekly downloads
 
Created
Source

travetto: Dependency Injection

Install: primary

$ npm install @travetto/di

Dependency injection is a framework primitive. When used in conjunction with automatic file scanning, it provides for handling of application dependency wiring. Due to the nature of typescript and type erasure of interfaces, dependency injection only supports classes as type signafiers. The primary goal of dependency injection is to allow for separation of concerns of object creation and it's usage.

Declaration

The @Injectable and @InjectableFactory decorators provide the registration of dependencies. Dependency declaration revolves around exposing classes and subtypes thereof to provide necessary functionality. Additionally, the framework will utilize dependencies to satisfy contracts with various backends (e.g. MongoModelSource provides itself as an injectable candidate for ModelSource).

Code: Example @Injectable

  @Injectable()
  class CustomService {
    async coolOperation() {
      ... work ...
    }
  }

When declaring a dependency, you can also provide a token to allow for multiple instances of the dependency to be defined. This can be used in many situations:

Code: Example @Injectable with multiple targets

  @Injectable()
  class CustomService {
    async coolOperation() {
      ... work ...
    }
  }

  const CUSTOM2 = Symbol('custom2');

  @Injectable({ target: CustomService, symbol: CUSTOM2 })
  class CustomService2 extends CustomService {
    async coolOperation() {
      await super.coolOperation();
      // Do some additional work
    }
  }

  class Consumer {
    @Inject(CUSTOM2) // Pull in specific service 
    service: CustomService;
  }

As you can see, the target field is also set, which indicates to the dependency registration process what class the injectable is compatible with. Additionally, when using abstract classes, the parent class is always considered as a valid candidate type.

Code: Example @Injectable with target via abstract class

  abstract class BaseService {
    abstract work():Promise<void>;
  }

  @Injectable()
  class SpecificService extends BaseService {
    async work() {
      // Do some additional work
    }
  }

In this scenario, SpecificService is a valid candidate for BaseService due to the abstract inheritance. Sometimes, you may want to provide a slight variation to a dependency without extending a class. To this end, the @InjectableFactory decorator denotes a static class method that produces an @Injectable.

Code: Example @InjectableFactory, return type defines target class

  class Config {
    @InjectableFactory()
    static initService(): CoolService {
      return new CoolService();
    }
  }

Given the static method initService, the function will be provided as a valid candidate for CoolService. Instead of calling the constructor of the type directly, this function will work as a factory for producing the injectable.

NOTE Due to the lack of typechecker in the Compiler for performance reasons, the return type on the factory method is mandatory. Without it, the code will not know what the expected target type should be.

NOTE Other modules are able to provide aliases to @Injectable that also provide additional functionality. For example, the @Config or the @Controller decorator registers the associated class as an injectable element.

Injection

Once all of your necessary dependencies are defined, now is the time to provide those @Injectable instances to your code. There are three primary methods for injection:

The @Inject decorator, which denotes a desire to inject a value directly. These will be set post construction.

Code: Example @Injectable with dependencies as @Inject fields

  @Injectable()
  class CustomService {
    @Inject()
    private dependentService: DependentService;
    
    async coolOperation() {
      await this.dependentService.doWork();
    }
  }

The @Injectable constructor params, which will be provided as the instance is being constructed.

Code: Example @Injectable with dependencies in constructor

  @Injectable()
  class CustomService {
    constructor (private dependentService: DependentService) {}
    
    async coolOperation() {
      await this.dependentService.doWork();
    }
  }

Via @InjectableFactory params, which are comparable to constructor params

Code: Example @InjectableFactory with parameters as dependencies

  class Config {
    @InjectableFactory()
    static initService(dependentService: DependentService): CustomService {
      return new CustomService(dependentService);
    }
  }

NOTE If you are building modules for others to consume, often times it is desirable to have default implementations for contracts that can be overridden. The @Inject decorator takes a flag of defaultIfMissing and a class name, that it will use to satisfy the dependency if nothing else matches. This would look like:

Code: Example of default if missing

  abstract class Contract {

  }

  @Injectable({ targets: SimpleContract })
  class SimpleContract extends Contract {}

  @Injectable({ targets: ComplexContract })
  class ComplexContract extends Contract {}

  @Injectable()
  class ContractConsumer {
    // Will default to SimpleContract if nothing else registered
    @Inject({ defaultIfMissing: SimpleContract })
    contract: Contract;
  }

  
  class Config {
    // Complex will be marked as the available Contract
    @InjectableFactory()
    getContract(complex: ComplexContract): Contract { 
      return complex;
    }
  }

Manual Invocation

Some times you will need to lookup a dependency dynamically, or you want to control the injection process at a more granular level. To achieve that you will need to directly access the DependencyRegistry. The registry allows for requesting a dependency by class reference:

Code: Example of manual lookup

@Injectable()
class Complex {}

class ManualLookup {
  async invoke() {
    const complex = await DependencyRegistry.getInstance(Complex);
  }
}

@Application

Given that dependency injection is generally a pre-requisite for application execution, it stands as the primary entrypoint for application invocation. The Base provides a simplistic bootstrap to allow for the application to run, but that is not sufficient for more complex applications.

The module provides a decorator, @Application who's job is to register entry points into the application. For example:

Code: Example of @Application target

import { Application, Inject, Injectable } from '../';

@Injectable()
class Server {
  name = 'roger';

  async launch() {
    ...
  }
}

@Application('simple')
class SimpleApp {

  @Inject()
  server: Server

  async run() {
    return this.server.launch();
  }
}

The @Application decorator exposes some additional functionality, which can be used to launch the application.

.run() Arguments

The arguments specified in the run method, will now be able to be specified when invoking the application from the command line. For instance:

Code: Simple Entrypoint with Parameters

@Application('simple')
class SimpleApp {
  async run(domain: string, port = 3000) {
    console.log('Launching', domain, 'on port', port);
  }
}

These command line invocation of travetto run would look like:

Terminal: Sample CLI Output

$ travetto run
....

     ● [e2e] simple
       ----------------------------------
       usage: simple [domain] (port=3000)

To invoke the simple application, you need to pass domain where port is optional with a default.

Terminal: Invoke Simple

$ travetto run simple mydomain.biz 4000

[INFO] Launching mydomain.biz on port 4000

Additionally, the parameters will be type checked, to ensure proper evaluation.

Terminal: Invoke Simple with bad port

$ travetto run simple mydomain.biz orange
usage: simple domain (string), port=[3000] (number)

The types are inferred from the .run() method parameters, but can be overridden in the @Application annotation to support customization. Only primitive types are supported:

  • number - Float or decimal
  • string - Default if no type is specified
  • boolean - true(yes/on/1) and false(no/off/0)
  • union - Type unions of the same type (string_a|string_b or 1|2|3|4)

Customizing the types is done by name, and allows for greater control:

Code: Complex Entrypoint with Customization

@Application('complex', {
  watchable: true,
  paramMap: {
    domain: {
      title: 'Domain Name',
      type: 'string',
      subtype: 'url'      
    },
    port : {
      title: 'Server Port',
      def: 3000
    }
  }
})
class ComplexApp {
  async run(domain: string, port: number) {
    console.log('Launching', domain, 'on port', port);
  }
}

*** Note *** The applications, by default, will not scan other application's folders. This means, if you have an application in the e2e/ folder, all of the code in your src/ folder will not be picked up automatically. This defined under the assumption that each application is unique. If you have an application that is an extension of the primary application (src/), you can specify the @Application config property of standalone to be false. This will now scan both folders to run your application.

Keywords

FAQs

Package last updated on 04 May 2019

Did you know?

Socket

Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.

Install

Related posts

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc